mirror of
https://github.com/SoftFever/OrcaSlicer.git
synced 2025-07-13 17:58:03 -06:00
PNGReadWrite: png::write_gray_to_file(), png::write_gray_to_file_scaled()
This commit is contained in:
parent
3985d7bb81
commit
032df7f18f
2 changed files with 72 additions and 15 deletions
|
@ -102,7 +102,8 @@ bool decode_png(IStream &in_buf, ImageGreyscale &out_img)
|
||||||
|
|
||||||
// Down to earth function to store a packed RGB image to file. Mostly useful for debugging purposes.
|
// Down to earth function to store a packed RGB image to file. Mostly useful for debugging purposes.
|
||||||
// Based on https://www.lemoda.net/c/write-png/
|
// Based on https://www.lemoda.net/c/write-png/
|
||||||
bool write_rgb_to_file(const char *file_name_utf8, size_t width, size_t height, const uint8_t *data_rgb)
|
// png_color_type is PNG_COLOR_TYPE_RGB or PNG_COLOR_TYPE_GRAY
|
||||||
|
static bool write_rgb_or_gray_to_file(const char *file_name_utf8, size_t width, size_t height, int png_color_type, const uint8_t *data)
|
||||||
{
|
{
|
||||||
bool result = false;
|
bool result = false;
|
||||||
|
|
||||||
|
@ -141,17 +142,20 @@ bool write_rgb_to_file(const char *file_name_utf8, size_t width, size_t height,
|
||||||
png_uint_32(width),
|
png_uint_32(width),
|
||||||
png_uint_32(height),
|
png_uint_32(height),
|
||||||
8, // depth
|
8, // depth
|
||||||
PNG_COLOR_TYPE_RGB,
|
png_color_type,
|
||||||
PNG_INTERLACE_NONE,
|
PNG_INTERLACE_NONE,
|
||||||
PNG_COMPRESSION_TYPE_DEFAULT,
|
PNG_COMPRESSION_TYPE_DEFAULT,
|
||||||
PNG_FILTER_TYPE_DEFAULT);
|
PNG_FILTER_TYPE_DEFAULT);
|
||||||
|
|
||||||
// Initialize rows of PNG.
|
// Initialize rows of PNG.
|
||||||
row_pointers = reinterpret_cast<png_byte**>(::png_malloc(png_ptr, height * sizeof(png_byte*)));
|
row_pointers = reinterpret_cast<png_byte**>(::png_malloc(png_ptr, height * sizeof(png_byte*)));
|
||||||
|
int line_width = width;
|
||||||
|
if (png_color_type == PNG_COLOR_TYPE_RGB)
|
||||||
|
line_width *= 3;
|
||||||
for (size_t y = 0; y < height; ++ y) {
|
for (size_t y = 0; y < height; ++ y) {
|
||||||
auto row = reinterpret_cast<png_byte*>(::png_malloc(png_ptr, sizeof(uint8_t) * width * 3));
|
auto row = reinterpret_cast<png_byte*>(::png_malloc(png_ptr, line_width));
|
||||||
row_pointers[y] = row;
|
row_pointers[y] = row;
|
||||||
memcpy(row, data_rgb + width * y * 3, sizeof(uint8_t) * width * 3);
|
memcpy(row, data + line_width * y, line_width);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Write the image data to "fp".
|
// Write the image data to "fp".
|
||||||
|
@ -174,6 +178,11 @@ fopen_failed:
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool write_rgb_to_file(const char *file_name_utf8, size_t width, size_t height, const uint8_t *data_rgb)
|
||||||
|
{
|
||||||
|
return write_rgb_or_gray_to_file(file_name_utf8, width, height, PNG_COLOR_TYPE_RGB, data_rgb);
|
||||||
|
}
|
||||||
|
|
||||||
bool write_rgb_to_file(const std::string &file_name_utf8, size_t width, size_t height, const uint8_t *data_rgb)
|
bool write_rgb_to_file(const std::string &file_name_utf8, size_t width, size_t height, const uint8_t *data_rgb)
|
||||||
{
|
{
|
||||||
return write_rgb_to_file(file_name_utf8.c_str(), width, height, data_rgb);
|
return write_rgb_to_file(file_name_utf8.c_str(), width, height, data_rgb);
|
||||||
|
@ -185,32 +194,54 @@ bool write_rgb_to_file(const std::string &file_name_utf8, size_t width, size_t h
|
||||||
return write_rgb_to_file(file_name_utf8.c_str(), width, height, data_rgb.data());
|
return write_rgb_to_file(file_name_utf8.c_str(), width, height, data_rgb.data());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool write_gray_to_file(const char *file_name_utf8, size_t width, size_t height, const uint8_t *data_gray)
|
||||||
|
{
|
||||||
|
return write_rgb_or_gray_to_file(file_name_utf8, width, height, PNG_COLOR_TYPE_GRAY, data_gray);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool write_gray_to_file(const std::string &file_name_utf8, size_t width, size_t height, const uint8_t *data_gray)
|
||||||
|
{
|
||||||
|
return write_gray_to_file(file_name_utf8.c_str(), width, height, data_gray);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool write_gray_to_file(const std::string &file_name_utf8, size_t width, size_t height, const std::vector<uint8_t> &data_gray)
|
||||||
|
{
|
||||||
|
assert(width * height == data_gray.size());
|
||||||
|
return write_gray_to_file(file_name_utf8.c_str(), width, height, data_gray.data());
|
||||||
|
}
|
||||||
|
|
||||||
// Scaled variants are mostly useful for debugging purposes, for example to export images of low resolution distance fileds.
|
// Scaled variants are mostly useful for debugging purposes, for example to export images of low resolution distance fileds.
|
||||||
// Scaling is done by multiplying rows and columns without any smoothing to emphasise the original pixels.
|
// Scaling is done by multiplying rows and columns without any smoothing to emphasise the original pixels.
|
||||||
bool write_rgb_to_file_scaled(const char *file_name_utf8, size_t width, size_t height, const uint8_t *data_rgb, size_t scale)
|
// png_color_type is PNG_COLOR_TYPE_RGB or PNG_COLOR_TYPE_GRAY
|
||||||
|
static bool write_rgb_or_gray_to_file_scaled(const char *file_name_utf8, size_t width, size_t height, int png_color_type, const uint8_t *data, size_t scale)
|
||||||
{
|
{
|
||||||
if (scale <= 1)
|
if (scale <= 1)
|
||||||
return write_rgb_to_file(file_name_utf8, width, height, data_rgb);
|
return write_rgb_or_gray_to_file(file_name_utf8, width, height, png_color_type, data);
|
||||||
else {
|
else {
|
||||||
std::vector<uint8_t> scaled(width * height * 3 * scale * scale);
|
size_t pixel_bytes = png_color_type == PNG_COLOR_TYPE_RGB ? 3 : 1;
|
||||||
|
size_t line_width = width * pixel_bytes;
|
||||||
|
std::vector<uint8_t> scaled(line_width * height * scale * scale);
|
||||||
uint8_t *dst = scaled.data();
|
uint8_t *dst = scaled.data();
|
||||||
for (size_t r = 0; r < height; ++ r) {
|
for (size_t r = 0; r < height; ++ r) {
|
||||||
for (size_t repr = 0; repr < scale; ++ repr) {
|
for (size_t repr = 0; repr < scale; ++ repr) {
|
||||||
const uint8_t *row = data_rgb + width * 3 * r;
|
const uint8_t *row = data + line_width * r;
|
||||||
for (size_t c = 0; c < width; ++ c) {
|
for (size_t c = 0; c < width; ++ c) {
|
||||||
for (size_t repc = 0; repc < scale; ++ repc) {
|
for (size_t repc = 0; repc < scale; ++ repc)
|
||||||
*dst ++ = row[0];
|
for (size_t b = 0; b < pixel_bytes; ++ b)
|
||||||
*dst ++ = row[1];
|
*dst ++ = row[b];
|
||||||
*dst ++ = row[2];
|
row += pixel_bytes;
|
||||||
}
|
|
||||||
row += 3;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return write_rgb_to_file(file_name_utf8, width * scale, height * scale, scaled.data());
|
return write_rgb_or_gray_to_file(file_name_utf8, width * scale, height * scale, png_color_type, scaled.data());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool write_rgb_to_file_scaled(const char *file_name_utf8, size_t width, size_t height, const uint8_t *data_rgb, size_t scale)
|
||||||
|
{
|
||||||
|
return write_rgb_or_gray_to_file_scaled(file_name_utf8, width, height, PNG_COLOR_TYPE_RGB, data_rgb, scale);
|
||||||
|
}
|
||||||
|
|
||||||
bool write_rgb_to_file_scaled(const std::string &file_name_utf8, size_t width, size_t height, const uint8_t *data_rgb, size_t scale)
|
bool write_rgb_to_file_scaled(const std::string &file_name_utf8, size_t width, size_t height, const uint8_t *data_rgb, size_t scale)
|
||||||
{
|
{
|
||||||
return write_rgb_to_file_scaled(file_name_utf8.c_str(), width, height, data_rgb, scale);
|
return write_rgb_to_file_scaled(file_name_utf8.c_str(), width, height, data_rgb, scale);
|
||||||
|
@ -222,4 +253,20 @@ bool write_rgb_to_file_scaled(const std::string &file_name_utf8, size_t width, s
|
||||||
return write_rgb_to_file_scaled(file_name_utf8.c_str(), width, height, data_rgb.data(), scale);
|
return write_rgb_to_file_scaled(file_name_utf8.c_str(), width, height, data_rgb.data(), scale);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool write_gray_to_file_scaled(const char *file_name_utf8, size_t width, size_t height, const uint8_t *data_gray, size_t scale)
|
||||||
|
{
|
||||||
|
return write_rgb_or_gray_to_file_scaled(file_name_utf8, width, height, PNG_COLOR_TYPE_GRAY, data_gray, scale);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool write_gray_to_file_scaled(const std::string &file_name_utf8, size_t width, size_t height, const uint8_t *data_gray, size_t scale)
|
||||||
|
{
|
||||||
|
return write_gray_to_file_scaled(file_name_utf8.c_str(), width, height, data_gray, scale);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool write_gray_to_file_scaled(const std::string &file_name_utf8, size_t width, size_t height, const std::vector<uint8_t> &data_gray, size_t scale)
|
||||||
|
{
|
||||||
|
assert(width * height == data_gray.size());
|
||||||
|
return write_gray_to_file_scaled(file_name_utf8.c_str(), width, height, data_gray.data(), scale);
|
||||||
|
}
|
||||||
|
|
||||||
}} // namespace Slic3r::png
|
}} // namespace Slic3r::png
|
||||||
|
|
|
@ -71,11 +71,21 @@ template<class Img> bool decode_png(const ReadBuf &in_buf, Img &out_img)
|
||||||
bool write_rgb_to_file(const char *file_name_utf8, size_t width, size_t height, const uint8_t *data_rgb);
|
bool write_rgb_to_file(const char *file_name_utf8, size_t width, size_t height, const uint8_t *data_rgb);
|
||||||
bool write_rgb_to_file(const std::string &file_name_utf8, size_t width, size_t height, const uint8_t *data_rgb);
|
bool write_rgb_to_file(const std::string &file_name_utf8, size_t width, size_t height, const uint8_t *data_rgb);
|
||||||
bool write_rgb_to_file(const std::string &file_name_utf8, size_t width, size_t height, const std::vector<uint8_t> &data_rgb);
|
bool write_rgb_to_file(const std::string &file_name_utf8, size_t width, size_t height, const std::vector<uint8_t> &data_rgb);
|
||||||
|
// Grayscale variants
|
||||||
|
bool write_gray_to_file(const char *file_name_utf8, size_t width, size_t height, const uint8_t *data_gray);
|
||||||
|
bool write_gray_to_file(const std::string &file_name_utf8, size_t width, size_t height, const uint8_t *data_gray);
|
||||||
|
bool write_gray_to_file(const std::string &file_name_utf8, size_t width, size_t height, const std::vector<uint8_t> &data_gray);
|
||||||
|
|
||||||
// Scaled variants are mostly useful for debugging purposes, for example to export images of low resolution distance fileds.
|
// Scaled variants are mostly useful for debugging purposes, for example to export images of low resolution distance fileds.
|
||||||
// Scaling is done by multiplying rows and columns without any smoothing to emphasise the original pixels.
|
// Scaling is done by multiplying rows and columns without any smoothing to emphasise the original pixels.
|
||||||
bool write_rgb_to_file_scaled(const char *file_name_utf8, size_t width, size_t height, const uint8_t *data_rgb, size_t scale);
|
bool write_rgb_to_file_scaled(const char *file_name_utf8, size_t width, size_t height, const uint8_t *data_rgb, size_t scale);
|
||||||
bool write_rgb_to_file_scaled(const std::string &file_name_utf8, size_t width, size_t height, const uint8_t *data_rgb, size_t scale);
|
bool write_rgb_to_file_scaled(const std::string &file_name_utf8, size_t width, size_t height, const uint8_t *data_rgb, size_t scale);
|
||||||
bool write_rgb_to_file_scaled(const std::string &file_name_utf8, size_t width, size_t height, const std::vector<uint8_t> &data_rgb, size_t scale);
|
bool write_rgb_to_file_scaled(const std::string &file_name_utf8, size_t width, size_t height, const std::vector<uint8_t> &data_rgb, size_t scale);
|
||||||
|
// Grayscale variants
|
||||||
|
bool write_gray_to_file_scaled(const char *file_name_utf8, size_t width, size_t height, const uint8_t *data_gray, size_t scale);
|
||||||
|
bool write_gray_to_file_scaled(const std::string &file_name_utf8, size_t width, size_t height, const uint8_t *data_gray, size_t scale);
|
||||||
|
bool write_gray_to_file_scaled(const std::string &file_name_utf8, size_t width, size_t height, const std::vector<uint8_t> &data_gray, size_t scale);
|
||||||
|
|
||||||
|
|
||||||
}} // namespace Slic3r::png
|
}} // namespace Slic3r::png
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue