170 lines
4.8 KiB
C++
170 lines
4.8 KiB
C++
|
|
||
|
#define _CRT_SECURE_NO_WARNINGS
|
||
|
|
||
|
#include "save-bmp.h"
|
||
|
|
||
|
#include <cstdio>
|
||
|
|
||
|
#ifdef MSDFGEN_USE_CPP11
|
||
|
#include <cstdint>
|
||
|
#else
|
||
|
typedef int int32_t;
|
||
|
typedef unsigned uint32_t;
|
||
|
typedef unsigned short uint16_t;
|
||
|
typedef unsigned char uint8_t;
|
||
|
#endif
|
||
|
|
||
|
#include "pixel-conversion.hpp"
|
||
|
|
||
|
namespace msdfgen {
|
||
|
|
||
|
template <typename T>
|
||
|
static bool writeValue(FILE *file, T value) {
|
||
|
#ifdef __BIG_ENDIAN__
|
||
|
T reverse = 0;
|
||
|
for (int i = 0; i < sizeof(T); ++i) {
|
||
|
reverse <<= 8;
|
||
|
reverse |= value&T(0xff);
|
||
|
value >>= 8;
|
||
|
}
|
||
|
return fwrite(&reverse, sizeof(T), 1, file) == 1;
|
||
|
#else
|
||
|
return fwrite(&value, sizeof(T), 1, file) == 1;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
static bool writeBmpHeader(FILE *file, int width, int height, int &paddedWidth) {
|
||
|
paddedWidth = (3*width+3)&~3;
|
||
|
const uint32_t bitmapStart = 54;
|
||
|
const uint32_t bitmapSize = paddedWidth*height;
|
||
|
const uint32_t fileSize = bitmapStart+bitmapSize;
|
||
|
|
||
|
writeValue<uint16_t>(file, 0x4d42u);
|
||
|
writeValue<uint32_t>(file, fileSize);
|
||
|
writeValue<uint16_t>(file, 0);
|
||
|
writeValue<uint16_t>(file, 0);
|
||
|
writeValue<uint32_t>(file, bitmapStart);
|
||
|
|
||
|
writeValue<uint32_t>(file, 40);
|
||
|
writeValue<int32_t>(file, width);
|
||
|
writeValue<int32_t>(file, height);
|
||
|
writeValue<uint16_t>(file, 1);
|
||
|
writeValue<uint16_t>(file, 24);
|
||
|
writeValue<uint32_t>(file, 0);
|
||
|
writeValue<uint32_t>(file, bitmapSize);
|
||
|
writeValue<uint32_t>(file, 2835);
|
||
|
writeValue<uint32_t>(file, 2835);
|
||
|
writeValue<uint32_t>(file, 0);
|
||
|
writeValue<uint32_t>(file, 0);
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
bool saveBmp(const BitmapConstRef<byte, 1> &bitmap, const char *filename) {
|
||
|
FILE *file = fopen(filename, "wb");
|
||
|
if (!file)
|
||
|
return false;
|
||
|
|
||
|
int paddedWidth;
|
||
|
writeBmpHeader(file, bitmap.width, bitmap.height, paddedWidth);
|
||
|
|
||
|
const uint8_t padding[4] = { };
|
||
|
int padLength = paddedWidth-3*bitmap.width;
|
||
|
for (int y = 0; y < bitmap.height; ++y) {
|
||
|
for (int x = 0; x < bitmap.width; ++x) {
|
||
|
uint8_t px = (uint8_t) *bitmap(x, y);
|
||
|
fwrite(&px, sizeof(uint8_t), 1, file);
|
||
|
fwrite(&px, sizeof(uint8_t), 1, file);
|
||
|
fwrite(&px, sizeof(uint8_t), 1, file);
|
||
|
}
|
||
|
fwrite(padding, 1, padLength, file);
|
||
|
}
|
||
|
|
||
|
return !fclose(file);
|
||
|
}
|
||
|
|
||
|
bool saveBmp(const BitmapConstRef<byte, 3> &bitmap, const char *filename) {
|
||
|
FILE *file = fopen(filename, "wb");
|
||
|
if (!file)
|
||
|
return false;
|
||
|
|
||
|
int paddedWidth;
|
||
|
writeBmpHeader(file, bitmap.width, bitmap.height, paddedWidth);
|
||
|
|
||
|
const uint8_t padding[4] = { };
|
||
|
int padLength = paddedWidth-3*bitmap.width;
|
||
|
for (int y = 0; y < bitmap.height; ++y) {
|
||
|
for (int x = 0; x < bitmap.width; ++x) {
|
||
|
uint8_t bgr[3] = {
|
||
|
(uint8_t) bitmap(x, y)[2],
|
||
|
(uint8_t) bitmap(x, y)[1],
|
||
|
(uint8_t) bitmap(x, y)[0]
|
||
|
};
|
||
|
fwrite(bgr, sizeof(uint8_t), 3, file);
|
||
|
}
|
||
|
fwrite(padding, 1, padLength, file);
|
||
|
}
|
||
|
|
||
|
return !fclose(file);
|
||
|
}
|
||
|
|
||
|
bool saveBmp(const BitmapConstRef<byte, 4> &bitmap, const char *filename) {
|
||
|
// RGBA not supported by the BMP format
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
bool saveBmp(const BitmapConstRef<float, 1> &bitmap, const char *filename) {
|
||
|
FILE *file = fopen(filename, "wb");
|
||
|
if (!file)
|
||
|
return false;
|
||
|
|
||
|
int paddedWidth;
|
||
|
writeBmpHeader(file, bitmap.width, bitmap.height, paddedWidth);
|
||
|
|
||
|
const uint8_t padding[4] = { };
|
||
|
int padLength = paddedWidth-3*bitmap.width;
|
||
|
for (int y = 0; y < bitmap.height; ++y) {
|
||
|
for (int x = 0; x < bitmap.width; ++x) {
|
||
|
uint8_t px = (uint8_t) pixelFloatToByte(*bitmap(x, y));
|
||
|
fwrite(&px, sizeof(uint8_t), 1, file);
|
||
|
fwrite(&px, sizeof(uint8_t), 1, file);
|
||
|
fwrite(&px, sizeof(uint8_t), 1, file);
|
||
|
}
|
||
|
fwrite(padding, 1, padLength, file);
|
||
|
}
|
||
|
|
||
|
return !fclose(file);
|
||
|
}
|
||
|
|
||
|
bool saveBmp(const BitmapConstRef<float, 3> &bitmap, const char *filename) {
|
||
|
FILE *file = fopen(filename, "wb");
|
||
|
if (!file)
|
||
|
return false;
|
||
|
|
||
|
int paddedWidth;
|
||
|
writeBmpHeader(file, bitmap.width, bitmap.height, paddedWidth);
|
||
|
|
||
|
const uint8_t padding[4] = { };
|
||
|
int padLength = paddedWidth-3*bitmap.width;
|
||
|
for (int y = 0; y < bitmap.height; ++y) {
|
||
|
for (int x = 0; x < bitmap.width; ++x) {
|
||
|
uint8_t bgr[3] = {
|
||
|
(uint8_t) pixelFloatToByte(bitmap(x, y)[2]),
|
||
|
(uint8_t) pixelFloatToByte(bitmap(x, y)[1]),
|
||
|
(uint8_t) pixelFloatToByte(bitmap(x, y)[0])
|
||
|
};
|
||
|
fwrite(bgr, sizeof(uint8_t), 3, file);
|
||
|
}
|
||
|
fwrite(padding, 1, padLength, file);
|
||
|
}
|
||
|
|
||
|
return !fclose(file);
|
||
|
}
|
||
|
|
||
|
bool saveBmp(const BitmapConstRef<float, 4> &bitmap, const char *filename) {
|
||
|
// RGBA not supported by the BMP format
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
}
|